home *** CD-ROM | disk | FTP | other *** search
/ Programming Languages Suite / ProgramD2.iso / Borland / Borland C++ V5.02 / WORDPAD.PAK / FORMATBA.CPP < prev    next >
C/C++ Source or Header  |  1997-05-06  |  21KB  |  804 lines

  1. // formatba.cpp : implementation file
  2. //
  3. // This is a part of the Microsoft Foundation Classes C++ library.
  4. // Copyright (C) 1992-1995 Microsoft Corporation
  5. // All rights reserved.
  6. //
  7. // This source code is only intended as a supplement to the
  8. // Microsoft Foundation Classes Reference and related
  9. // electronic documentation provided with the library.
  10. // See these sources for detailed information regarding the
  11. // Microsoft Foundation Classes product.
  12.  
  13. #include "stdafx.h"
  14.  
  15. #include "wordpad.h"
  16. #include "wordpdoc.h"
  17. #include "wordpvw.h"
  18. #include "formatba.h"
  19. #include "strings.h"
  20.  
  21. #ifdef _DEBUG
  22. #undef THIS_FILE
  23. static char BASED_CODE THIS_FILE[] = __FILE__;
  24. #endif
  25.  
  26. // reserve lobyte for charset
  27. #define PRINTER_FONT 0x0100
  28. #define TT_FONT    0x0200
  29. #define DEVICE_FONT 0x0400
  30.  
  31. #define BMW 16
  32. #define BMH 15
  33.  
  34. static int nFontSizes[] = 
  35.     {8, 9, 10, 11, 12, 14, 16, 18, 20, 22, 24, 26, 28, 36, 48, 72};
  36. int CLocalComboBox::m_nFontHeight = 0;
  37.  
  38. class CFontDesc
  39. {
  40. public:
  41.     CFontDesc(LPCTSTR lpszName, LPCTSTR lpszScript, BYTE nCharSet, 
  42.         BYTE nPitchAndFamily, DWORD dwFlags);
  43.     CString m_strName;
  44.     CString m_strScript;
  45.     BYTE m_nCharSet;
  46.     BYTE m_nPitchAndFamily;
  47.     DWORD m_dwFlags;
  48. };
  49.  
  50. CFontDesc::CFontDesc(LPCTSTR lpszName, LPCTSTR lpszScript, BYTE nCharSet, 
  51.     BYTE nPitchAndFamily, DWORD dwFlags)
  52. {
  53.     m_strName = lpszName;
  54.     m_strScript = lpszScript;
  55.     m_nCharSet = nCharSet;
  56.     m_nPitchAndFamily = nPitchAndFamily;
  57.     m_dwFlags = dwFlags;
  58. }
  59.  
  60. BEGIN_MESSAGE_MAP(CFormatBar, CToolBar)
  61.     //{{AFX_MSG_MAP(CFormatBar)
  62.     ON_WM_CREATE()
  63.     ON_WM_DESTROY()
  64.     //}}AFX_MSG_MAP
  65.     ON_CBN_DROPDOWN(IDC_FONTSIZE, OnFontSizeDropDown)
  66.     ON_CBN_KILLFOCUS(IDC_FONTNAME, OnFontNameKillFocus)
  67.     ON_CBN_KILLFOCUS(IDC_FONTSIZE, OnFontSizeKillFocus)
  68.     ON_CBN_SETFOCUS(IDC_FONTNAME, OnComboSetFocus)
  69.     ON_CBN_SETFOCUS(IDC_FONTSIZE, OnComboSetFocus)
  70.     ON_CBN_CLOSEUP(IDC_FONTNAME, OnComboCloseUp)
  71.     ON_CBN_CLOSEUP(IDC_FONTSIZE, OnComboCloseUp)
  72.     ON_REGISTERED_MESSAGE(CWordPadApp::m_nPrinterChangedMsg, OnPrinterChanged)
  73.     // Global help commands
  74. END_MESSAGE_MAP()
  75.  
  76. static CSize GetBaseUnits(CFont* pFont)
  77. {
  78.     ASSERT(pFont != NULL);
  79.     ASSERT(pFont->GetSafeHandle() != NULL);
  80.     pFont = theApp.m_dcScreen.SelectObject(pFont);
  81.     TEXTMETRIC tm;
  82.     VERIFY(theApp.m_dcScreen.GetTextMetrics(&tm));
  83.  
  84.     theApp.m_dcScreen.SelectObject(pFont);
  85. //    return CSize(tm.tmAveCharWidth, tm.tmHeight+tm.tmDescent);
  86.     return CSize(tm.tmAveCharWidth, tm.tmHeight);
  87. }
  88.  
  89. CFormatBar::CFormatBar()
  90. {
  91.     CFont fnt;
  92.       fnt.Attach(GetStockObject(theApp.m_nDefFont));
  93.     m_szBaseUnits = GetBaseUnits(&fnt);
  94.     CLocalComboBox::m_nFontHeight = m_szBaseUnits.cy;
  95. }
  96.  
  97. void CFormatBar::OnUpdateCmdUI(CFrameWnd* pTarget, BOOL bDisableIfNoHndler)
  98. {
  99.     CToolBar::OnUpdateCmdUI(pTarget, bDisableIfNoHndler);
  100.     // don't update combo boxes if either one has the focus
  101.     if (!m_comboFontName.HasFocus() && !m_comboFontSize.HasFocus())
  102.         SyncToView();
  103. }
  104.  
  105. void CFormatBar::SyncToView()
  106. {
  107.     USES_CONVERSION;
  108.     // get the current font from the view and update
  109.     CHARHDR fh;
  110.     CHARFORMAT& cf = fh.cf;
  111.     fh.hwndFrom = m_hWnd;
  112.     fh.idFrom = GetDlgCtrlID();
  113.     fh.code = FN_GETFORMAT;
  114.     VERIFY(GetOwner()->SendMessage(WM_NOTIFY, fh.idFrom, (LPARAM)&fh));
  115.  
  116.     // the selection must be same font and charset to display correctly
  117.     if ((cf.dwMask & (CFM_FACE|CFM_CHARSET)) == (CFM_FACE|CFM_CHARSET))
  118.         m_comboFontName.MatchFont(A2T(cf.szFaceName), cf.bCharSet);
  119.     else
  120.         m_comboFontName.SetTheText(_T(""));
  121.  
  122.     // SetTwipSize only updates if different
  123.     // -1 means selection is not a single point size
  124.     m_comboFontSize.SetTwipSize( (cf.dwMask & CFM_SIZE) ? cf.yHeight : -1);
  125. }
  126.  
  127. void CFormatBar::OnFontSizeDropDown()
  128. {
  129.     CString str;
  130.     m_comboFontName.GetTheText(str);
  131.     LPCTSTR lpszName = NULL;
  132.     BOOL bPrinterFont;
  133.     int nIndex = m_comboFontName.FindStringExact(-1, str);
  134.     if (nIndex != CB_ERR)
  135.     {
  136.         CFontDesc* pDesc = (CFontDesc*)m_comboFontName.GetItemData(nIndex);
  137.         ASSERT(pDesc != NULL);
  138.         bPrinterFont = pDesc->m_dwFlags & PRINTER_FONT;
  139.         lpszName = pDesc->m_strName;
  140.     }
  141.  
  142.     int nSize = m_comboFontSize.GetTwipSize();
  143.     if (nSize == -2) // error
  144.     {
  145.         AfxMessageBox(IDS_INVALID_NUMBER, MB_OK|MB_ICONINFORMATION);
  146.         nSize = m_comboFontSize.m_nTwipsLast;
  147.     }
  148.     else if ((nSize >= 0 && nSize < 20) || nSize > 32760)
  149.     {
  150.         AfxMessageBox(IDS_INVALID_FONTSIZE, MB_OK|MB_ICONINFORMATION);
  151.         nSize = m_comboFontSize.m_nTwipsLast;
  152.     }
  153.  
  154.     if (bPrinterFont)
  155.         m_comboFontSize.EnumFontSizes(m_dcPrinter, lpszName);
  156.     else
  157.         m_comboFontSize.EnumFontSizes(theApp.m_dcScreen, lpszName);
  158.     m_comboFontSize.SetTwipSize(nSize);
  159. }
  160.  
  161. void CFormatBar::OnComboCloseUp()
  162. {
  163.     NotifyOwner(NM_RETURN);
  164. }
  165.  
  166. void CFormatBar::OnComboSetFocus()
  167. {
  168.     NotifyOwner(NM_SETFOCUS);
  169. }
  170.  
  171. void CFormatBar::OnFontNameKillFocus()
  172. {
  173.     USES_CONVERSION;
  174.     // get the current font from the view and update
  175.     NotifyOwner(NM_KILLFOCUS);
  176.  
  177.     CCharFormat cf;
  178.     cf.szFaceName[0] = NULL;
  179.  
  180.     // this will retrieve the font entered in the edit control
  181.     // it tries to match the font to something already present in the combo box
  182.     // this effectively ignores case of a font the user enters
  183.     // if a user enters arial, this will cause it to become Arial
  184.     CString str;
  185.     m_comboFontName.GetTheText(str);    // returns "arial"
  186.     m_comboFontName.SetTheText(str);                    // selects "Arial"
  187.     m_comboFontName.GetTheText(str);    // returns "Arial"
  188.  
  189.     // if font name box is not empty
  190.     if (str[0] != NULL)
  191.     {
  192.         cf.dwMask = CFM_FACE | CFM_CHARSET;
  193.         int nIndex = m_comboFontName.FindStringExact(-1, str);
  194.         if (nIndex != CB_ERR)
  195.         {
  196.             CFontDesc* pDesc = (CFontDesc*)m_comboFontName.GetItemData(nIndex);
  197.             ASSERT(pDesc != NULL);
  198.             ASSERT(pDesc->m_strName.GetLength() < LF_FACESIZE);
  199.             lstrcpynA(cf.szFaceName, T2A(pDesc->m_strName), LF_FACESIZE);
  200.             cf.bCharSet = pDesc->m_nCharSet;
  201.             cf.bPitchAndFamily = pDesc->m_nPitchAndFamily;
  202.         }
  203.         else // unknown font
  204.         {
  205.             ASSERT(str.GetLength() < LF_FACESIZE);
  206.             lstrcpynA(cf.szFaceName, T2A(str), LF_FACESIZE);
  207.             cf.bCharSet = DEFAULT_CHARSET;
  208.             cf.bPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE;
  209.         }
  210.         SetCharFormat(cf);
  211.     }
  212. }
  213.  
  214. void CFormatBar::OnFontSizeKillFocus()
  215. {
  216.     NotifyOwner(NM_KILLFOCUS);
  217.     int nSize = m_comboFontSize.GetTwipSize();
  218.     if (nSize == -2)
  219.     {
  220.         AfxMessageBox(IDS_INVALID_NUMBER, MB_OK|MB_ICONINFORMATION);
  221.         nSize = m_comboFontSize.m_nTwipsLast;
  222.     }
  223.     else if ((nSize >= 0 && nSize < 20) || nSize > 32760)
  224.     {
  225.         AfxMessageBox(IDS_INVALID_FONTSIZE, MB_OK|MB_ICONINFORMATION);
  226.         nSize = m_comboFontSize.m_nTwipsLast;
  227.     }
  228.     else if (nSize > 0)
  229.     {
  230.         CCharFormat cf;
  231.         cf.dwMask = CFM_SIZE;
  232.         cf.yHeight = nSize;
  233.         SetCharFormat(cf);
  234.     }
  235. }
  236.  
  237. LONG CFormatBar::OnPrinterChanged(UINT, LONG)
  238. {
  239.     theApp.CreatePrinterDC(m_dcPrinter);
  240.     m_comboFontName.EnumFontFamiliesEx(m_dcPrinter);
  241.     return 0;
  242. }
  243.  
  244. int CFormatBar::OnCreate(LPCREATESTRUCT lpCreateStruct)
  245. {
  246.     if (CToolBar::OnCreate(lpCreateStruct) == -1)
  247.         return -1;
  248.  
  249.     theApp.m_listPrinterNotify.AddTail(m_hWnd);
  250.     
  251.     CRect rect(0,0, (3*LF_FACESIZE*m_szBaseUnits.cx)/2, 200);
  252.     if (!m_comboFontName.Create(WS_TABSTOP|WS_VISIBLE|WS_TABSTOP|
  253.         WS_VSCROLL|CBS_DROPDOWN|CBS_SORT|CBS_AUTOHSCROLL|CBS_HASSTRINGS|
  254.         CBS_OWNERDRAWFIXED, rect, this, IDC_FONTNAME))
  255.     {
  256.         TRACE0("Failed to create fontname combo-box\n");
  257.         return -1;
  258.     }
  259.     m_comboFontName.LimitText(LF_FACESIZE);
  260.  
  261.     rect.SetRect(0, 0, 10*m_szBaseUnits.cx, 200);
  262.     if (!m_comboFontSize.Create(WS_TABSTOP|WS_VISIBLE|WS_TABSTOP|
  263.         WS_VSCROLL|CBS_DROPDOWN, rect, this, IDC_FONTSIZE))
  264.     {
  265.         TRACE0("Failed to create fontsize combo-box\n");
  266.         return -1;
  267.     }
  268.  
  269.     m_comboFontSize.LimitText(4);
  270.     m_comboFontName.EnumFontFamiliesEx(m_dcPrinter);
  271.     
  272.     return 0;
  273. }
  274.  
  275. void CFormatBar::OnDestroy() 
  276. {
  277.     CToolBar::OnDestroy();
  278.     POSITION pos = theApp.m_listPrinterNotify.Find(m_hWnd);
  279.     ASSERT(pos != NULL);
  280.     theApp.m_listPrinterNotify.RemoveAt(pos);
  281. }
  282.  
  283. void CFormatBar::PositionCombos()
  284. {
  285.     CRect rect;
  286.     // make font name box same size as font size box
  287.     // this is necessary since font name box is owner draw
  288.     m_comboFontName.SetItemHeight(-1, m_comboFontSize.GetItemHeight(-1));
  289.  
  290.     m_comboFontName.GetWindowRect(&rect);
  291.     int nHeight = rect.Height();
  292.  
  293.     m_comboFontName.GetWindowRect(&rect);
  294.     SetButtonInfo(0, IDC_FONTNAME, TBBS_SEPARATOR, rect.Width());
  295.     GetItemRect(0, &rect); // FontName ComboBox
  296.     m_comboFontName.SetWindowPos(NULL, rect.left, 
  297.         ((rect.Height() - nHeight) / 2) + rect.top, 0, 0, 
  298.         SWP_NOZORDER|SWP_NOSIZE|SWP_NOACTIVATE);
  299.  
  300.     m_comboFontSize.GetWindowRect(&rect);
  301.     SetButtonInfo(2, IDC_FONTSIZE, TBBS_SEPARATOR, rect.Width());
  302.     GetItemRect(2, &rect); // FontSize ComboBox
  303.     m_comboFontSize.SetWindowPos(NULL, rect.left, 
  304.         ((rect.Height() - nHeight) / 2) + rect.top, 0, 0, 
  305.         SWP_NOZORDER|SWP_NOSIZE|SWP_NOACTIVATE);
  306. }
  307.  
  308. /////////////////////////////////////////////////////////////////////////////
  309. // CFontComboBox 
  310.  
  311. BEGIN_MESSAGE_MAP(CFontComboBox, CLocalComboBox)
  312.     //{{AFX_MSG_MAP(CFontComboBox)
  313.     ON_WM_DESTROY()
  314.     //}}AFX_MSG_MAP
  315.     // Global help commands
  316. END_MESSAGE_MAP()
  317.  
  318. CFontComboBox::CFontComboBox()
  319. {
  320.     VERIFY(m_bmFontType.LoadBitmap(IDB_FONTTYPE));
  321. }
  322.  
  323. void CFontComboBox::OnDestroy() 
  324. {
  325.     // destroy all the CFontDesc's
  326.     EmptyContents();
  327.     CLocalComboBox::OnDestroy();
  328. }
  329.  
  330. void CFontComboBox::EmptyContents()
  331. {
  332.     // destroy all the CFontDesc's
  333.     int nCount = GetCount();
  334.     for (int i=0;i<nCount;i++)
  335.         delete (CFontDesc*)GetItemData(i);
  336. }
  337.  
  338. void CFontComboBox::EnumFontFamiliesEx(CDC& dc, BYTE nCharSet)
  339. {
  340.     CMapStringToPtr map;
  341.     CString str;
  342.     GetTheText(str);
  343.  
  344.     EmptyContents();
  345.     ResetContent();
  346.     LOGFONT lf;
  347.     memset(&lf, 0, sizeof(LOGFONT));
  348.     lf.lfCharSet = nCharSet;
  349.  
  350.     if (dc.m_hDC != NULL)
  351.     {
  352.         if (theApp.m_bWin4)
  353.         {
  354.             ::EnumFontFamiliesEx(dc.m_hDC, &lf,
  355.                 (FONTENUMPROC) EnumFamPrinterCallBackEx, (LPARAM) this, NULL);
  356.         }
  357.         else
  358.         {
  359.             ::EnumFontFamilies(dc.m_hDC, NULL,
  360.                 (FONTENUMPROC) EnumFamPrinterCallBack, (LPARAM) this);
  361.         }
  362.     }
  363.     else
  364.     {
  365.         HDC hDC = theApp.m_dcScreen.m_hDC;
  366.         ASSERT(hDC != NULL);
  367.         if (theApp.m_bWin4)
  368.         {
  369.             ::EnumFontFamiliesEx(hDC, &lf,
  370.                 (FONTENUMPROC) EnumFamScreenCallBackEx, (LPARAM) this, NULL);
  371.         }
  372.         else
  373.         {
  374.             ::EnumFontFamilies(hDC, NULL,
  375.                 (FONTENUMPROC) EnumFamScreenCallBack, (LPARAM) this);
  376.         }
  377.     }
  378.     // now walk through the fonts and remove (charset) from fonts with only one
  379.  
  380.     int nCount = m_arrayFontDesc.GetSize();
  381.     // walk through fonts adding names to string map
  382.     // first time add value 0, after that add value 1
  383.     for (int i = 0; i<nCount;i++)
  384.     {
  385.         CFontDesc* pDesc = (CFontDesc*)m_arrayFontDesc[i];
  386.         void* pv = NULL;
  387.         if (map.Lookup(pDesc->m_strName, pv)) // found it
  388.         {
  389.             if (pv == NULL) // only one entry so far
  390.             {
  391.                 map.RemoveKey(pDesc->m_strName);
  392.                 map.SetAt(pDesc->m_strName, (void*)1);
  393.             }
  394.         }
  395.         else // not found
  396.             map.SetAt(pDesc->m_strName, (void*)0);
  397.     }
  398.  
  399.     for (i = 0; i<nCount;i++)
  400.     {
  401.         CFontDesc* pDesc = (CFontDesc*)m_arrayFontDesc[i];
  402.         CString str = pDesc->m_strName;
  403.         void* pv = NULL;
  404.         VERIFY(map.Lookup(str, pv));
  405.         if (pv != NULL && !pDesc->m_strScript.IsEmpty())
  406.         {
  407.             str += " (";
  408.             str += pDesc->m_strScript;
  409.             str += ")";
  410.         }
  411.  
  412.         int nIndex = AddString(str);
  413.         ASSERT(nIndex >=0);
  414.         if (nIndex >=0) //no error
  415.             SetItemData(nIndex, (DWORD)pDesc);
  416.     }
  417.  
  418.     SetTheText(str);
  419.     m_arrayFontDesc.RemoveAll();
  420. }
  421.  
  422. void CFontComboBox::AddFont(ENUMLOGFONT* pelf, DWORD dwType, LPCTSTR lpszScript)
  423. {
  424.     LOGFONT& lf = pelf->elfLogFont;
  425.     if (lf.lfCharSet == MAC_CHARSET) // don't put in MAC fonts, commdlg doesn't either
  426.         return;
  427.     // Don't display vertical font for FE platform
  428.     if ((GetSystemMetrics(SM_DBCSENABLED)) && (lf.lfFaceName[0] == '@'))
  429.         return;
  430.     // don't put in non-printer raster fonts
  431.     CFontDesc* pDesc = new CFontDesc(lf.lfFaceName, lpszScript, 
  432.         lf.lfCharSet, lf.lfPitchAndFamily, dwType);
  433.     m_arrayFontDesc.Add(pDesc);
  434. }
  435.  
  436. BOOL CALLBACK AFX_EXPORT CFontComboBox::EnumFamScreenCallBack(ENUMLOGFONT* pelf, 
  437.     NEWTEXTMETRICEX* /*lpntm*/, int FontType, LPVOID pThis)
  438. {
  439.     // don't put in non-printer raster fonts
  440.     if (FontType & RASTER_FONTTYPE)
  441.         return 1;
  442.     DWORD dwData = (FontType & TRUETYPE_FONTTYPE) ? TT_FONT : 0;
  443.     ((CFontComboBox *)pThis)->AddFont(pelf, dwData);
  444.     return 1;
  445. }
  446.  
  447. BOOL CALLBACK AFX_EXPORT CFontComboBox::EnumFamPrinterCallBack(ENUMLOGFONT* pelf, 
  448.     NEWTEXTMETRICEX* /*lpntm*/, int FontType, LPVOID pThis)
  449. {
  450.     DWORD dwData = PRINTER_FONT;
  451.     if (FontType & TRUETYPE_FONTTYPE)
  452.         dwData |= TT_FONT;
  453.     else if (FontType & DEVICE_FONTTYPE)
  454.         dwData |= DEVICE_FONT;
  455.     ((CFontComboBox *)pThis)->AddFont(pelf, dwData);
  456.     return 1;
  457. }
  458.  
  459. BOOL CALLBACK AFX_EXPORT CFontComboBox::EnumFamScreenCallBackEx(ENUMLOGFONTEX* pelf, 
  460.     NEWTEXTMETRICEX* /*lpntm*/, int FontType, LPVOID pThis)
  461. {
  462.     // don't put in non-printer raster fonts
  463.     if (FontType & RASTER_FONTTYPE)
  464.         return 1;
  465.     DWORD dwData = (FontType & TRUETYPE_FONTTYPE) ? TT_FONT : 0;
  466.     ((CFontComboBox *)pThis)->AddFont((ENUMLOGFONT*)pelf, dwData, CString(pelf->elfScript));
  467.     return 1;
  468. }
  469.  
  470. BOOL CALLBACK AFX_EXPORT CFontComboBox::EnumFamPrinterCallBackEx(ENUMLOGFONTEX* pelf, 
  471.     NEWTEXTMETRICEX* /*lpntm*/, int FontType, LPVOID pThis)
  472. {
  473.     DWORD dwData = PRINTER_FONT;
  474.     if (FontType & TRUETYPE_FONTTYPE)
  475.         dwData |= TT_FONT;
  476.     else if (FontType & DEVICE_FONTTYPE)
  477.         dwData |= DEVICE_FONT;
  478.     ((CFontComboBox *)pThis)->AddFont((ENUMLOGFONT*)pelf, dwData, CString(pelf->elfScript));
  479.     return 1;
  480. }
  481.  
  482. void CFontComboBox::DrawItem(LPDRAWITEMSTRUCT lpDIS)
  483. {
  484.     ASSERT(lpDIS->CtlType == ODT_COMBOBOX);
  485.     int id = (int)(WORD)lpDIS->itemID;
  486.  
  487.     CDC *pDC = CDC::FromHandle(lpDIS->hDC);
  488.     CRect rc(lpDIS->rcItem);
  489.     if (lpDIS->itemState & ODS_FOCUS)
  490.         pDC->DrawFocusRect(rc);
  491.     int nIndexDC = pDC->SaveDC();
  492.  
  493.     CBrush brushFill;
  494.     if (lpDIS->itemState & ODS_SELECTED)
  495.     {
  496.         brushFill.CreateSolidBrush(::GetSysColor(COLOR_HIGHLIGHT));
  497.         pDC->SetTextColor(::GetSysColor(COLOR_HIGHLIGHTTEXT));
  498.     }
  499.     else
  500.         brushFill.CreateSolidBrush(pDC->GetBkColor());
  501.     pDC->SetBkMode(TRANSPARENT);
  502.     pDC->FillRect(rc, &brushFill);
  503.  
  504.     CFontDesc* pDesc= (CFontDesc*)lpDIS->itemData;
  505.     ASSERT(pDesc != NULL);
  506.     DWORD dwData = pDesc->m_dwFlags;
  507.     if (dwData & (TT_FONT|DEVICE_FONT)) // truetype or device flag set by SetItemData
  508.     {
  509.         CDC dc;
  510.         dc.CreateCompatibleDC(pDC);
  511.         CBitmap* pBitmap = dc.SelectObject(&m_bmFontType);
  512.         if (dwData & TT_FONT)
  513.             pDC->BitBlt(rc.left, rc.top, BMW, BMH, &dc, BMW, 0, SRCAND);
  514.         else // DEVICE_FONT
  515.             pDC->BitBlt(rc.left, rc.top, BMW, BMH, &dc, 0, 0, SRCAND);
  516.         dc.SelectObject(pBitmap);
  517.     }
  518.     
  519.     rc.left += BMW + 6;
  520.     CString strText;
  521.     GetLBText(id, strText);
  522.     pDC->TextOut(rc.left,rc.top,strText,strText.GetLength());
  523.  
  524.     pDC->RestoreDC(nIndexDC);
  525. }
  526.  
  527. void CFontComboBox::MeasureItem(LPMEASUREITEMSTRUCT lpMIS)
  528. {
  529.     ASSERT(lpMIS->CtlType == ODT_COMBOBOX);
  530.     ASSERT(m_nFontHeight > 0);
  531.     CRect rc;
  532.     
  533.     GetWindowRect(&rc);
  534.     lpMIS->itemWidth = rc.Width();
  535.     lpMIS->itemHeight = max(BMH, m_nFontHeight);
  536. }
  537.  
  538. int CFontComboBox::CompareItem(LPCOMPAREITEMSTRUCT lpCIS)
  539. {
  540.     ASSERT(lpCIS->CtlType == ODT_COMBOBOX);
  541.     int id1 = (int)(WORD)lpCIS->itemID1;
  542.     int id2 = (int)(WORD)lpCIS->itemID2;
  543.     CString str1,str2;
  544.     if (id1 == -1)
  545.         return -1;
  546.     if (id2 == -1)
  547.         return 1;
  548.     GetLBText(id1, str1);
  549.     GetLBText(id2, str2);
  550.     return str1.Collate(str2);
  551. }
  552.  
  553. // find a font with the face name and charset
  554. void CFontComboBox::MatchFont(LPCTSTR lpszName, BYTE nCharSet)
  555. {
  556.     int nFirstIndex = FindString(-1, lpszName);
  557.     if (nFirstIndex != CB_ERR)
  558.     {
  559.         int nIndex = nFirstIndex;
  560.         do
  561.         { 
  562.             CFontDesc* pDesc = (CFontDesc*)GetItemData(nIndex);
  563.             ASSERT(pDesc != NULL);
  564.             // check the actual font name to avoid matching Courier western
  565.             // to Courier New western
  566.             if ((nCharSet == DEFAULT_CHARSET || pDesc->m_nCharSet == nCharSet) && 
  567.                 lstrcmp(lpszName, pDesc->m_strName)==0)
  568.             {
  569.                 //got a match
  570.                 if (GetCurSel() != nIndex)
  571.                     SetCurSel(nIndex);
  572.                 return;
  573.             }
  574.             nIndex = FindString(nIndex, lpszName);
  575.         } while (nIndex != nFirstIndex);
  576.         // loop until found or back to first item again
  577.     }
  578.     //enter font name
  579.     SetTheText(lpszName, TRUE);
  580. }
  581.  
  582. /////////////////////////////////////////////////////////////////////////////
  583. // CSizeComboBox 
  584.  
  585. CSizeComboBox::CSizeComboBox()
  586. {
  587.     m_nTwipsLast = 0;
  588. }
  589.  
  590. void CSizeComboBox::EnumFontSizes(CDC& dc, LPCTSTR pFontName)
  591. {
  592.     ResetContent();
  593.     if (pFontName == NULL)
  594.         return;
  595.     if (pFontName[0] == NULL)
  596.         return;
  597.     
  598.     ASSERT(dc.m_hDC != NULL);
  599.     m_nLogVert = dc.GetDeviceCaps(LOGPIXELSY);
  600.  
  601.     ::EnumFontFamilies(dc.m_hDC, pFontName,
  602.         (FONTENUMPROC) EnumSizeCallBack, (LPARAM) this);
  603. }
  604.  
  605. void CSizeComboBox::TwipsToPointString(LPTSTR lpszBuf, int nTwips)
  606. {
  607.     ASSERT(lpszBuf != NULL);
  608.     lpszBuf[0] = NULL;
  609.     if (nTwips >= 0)
  610.     {
  611.         // round to nearest half point
  612.         nTwips = (nTwips+5)/10;
  613.         if ((nTwips%2) == 0)
  614.             _stprintf(lpszBuf, _T("%ld"), nTwips/2);
  615.         else
  616.             _stprintf(lpszBuf, _T("%.1f"), (float)nTwips/2.F);
  617.     }
  618. }
  619.  
  620. void CSizeComboBox::SetTwipSize(int nTwips)
  621. {
  622.     if (nTwips != GetTwipSize())
  623.     {
  624.         TCHAR buf[10];
  625.         TwipsToPointString(buf, nTwips);
  626.         SetTheText(buf, TRUE);
  627.     }
  628.     m_nTwipsLast = nTwips;
  629. }
  630.  
  631. int CSizeComboBox::GetTwipSize()
  632. {
  633.     // return values
  634.     // -2 -- error
  635.     // -1 -- edit box empty
  636.     // >=0 -- font size in twips
  637.     CString str;
  638.     GetTheText(str);
  639.     LPCTSTR lpszText = str;
  640.  
  641.     while (*lpszText == ' ' || *lpszText == '\t')
  642.         lpszText++;
  643.  
  644.     if (lpszText[0] == NULL)
  645.         return -1; // no text in control
  646.  
  647.     double d = _tcstod(lpszText, (LPTSTR*)&lpszText);
  648.     while (*lpszText == ' ' || *lpszText == '\t')
  649.         lpszText++;
  650.  
  651.     if (*lpszText != NULL)
  652.         return -2;   // not terminated properly
  653.  
  654.     return (d<0.) ? 0 : (int)(d*20.);
  655. }
  656.  
  657. BOOL CALLBACK AFX_EXPORT CSizeComboBox::EnumSizeCallBack(LOGFONT FAR* /*lplf*/, 
  658.         LPNEWTEXTMETRIC lpntm, int FontType, LPVOID lpv)
  659. {
  660.     CSizeComboBox* pThis = (CSizeComboBox*)lpv;
  661.     ASSERT(pThis != NULL);
  662.     TCHAR buf[10];
  663.     if (
  664.         (FontType & TRUETYPE_FONTTYPE) || 
  665.         !( (FontType & TRUETYPE_FONTTYPE) || (FontType & RASTER_FONTTYPE) )
  666.         ) // if truetype or vector font
  667.     {
  668.         // this occurs when there is a truetype and nontruetype version of a font
  669.         if (pThis->GetCount() != 0)
  670.             pThis->ResetContent();
  671.                     
  672.         for (int i = 0; i < 16; i++)
  673.         {
  674.             wsprintf(buf, _T("%d"), nFontSizes[i]);
  675.             pThis->AddString(buf);
  676.         }
  677.         return FALSE; // don't call me again
  678.     }
  679.     // calc character height in pixels
  680.     pThis->InsertSize(MulDiv(lpntm->tmHeight-lpntm->tmInternalLeading, 
  681.         1440, pThis->m_nLogVert));
  682.     return TRUE; // call me again
  683. }
  684.  
  685. void CSizeComboBox::InsertSize(int nSize)
  686. {
  687.     ASSERT(nSize > 0);
  688.     DWORD dwSize = (DWORD)nSize;
  689.     TCHAR buf[10];
  690.     TwipsToPointString(buf, nSize);
  691.     if (FindStringExact(-1, buf) == CB_ERR)
  692.     {
  693.         int nIndex = -1;
  694.         int nPos = 0;
  695.         DWORD dw;
  696.         while ((dw = GetItemData(nPos)) != CB_ERR)
  697.         {
  698.             if (dw > dwSize)
  699.             {
  700.                 nIndex = nPos;
  701.                 break;
  702.             }
  703.             nPos++;
  704.         }
  705.         nIndex = InsertString(nIndex, buf);
  706.         ASSERT(nIndex != CB_ERR);
  707.         if (nIndex != CB_ERR)
  708.             SetItemData(nIndex, dwSize);
  709.     }
  710. }
  711.  
  712. /////////////////////////////////////////////////////////////////////////////
  713. // CLocalComboBox 
  714.  
  715. BEGIN_MESSAGE_MAP(CLocalComboBox, CComboBox)
  716.     //{{AFX_MSG_MAP(CLocalComboBox)
  717.     ON_WM_CREATE()
  718.     //}}AFX_MSG_MAP
  719.     // Global help commands
  720. END_MESSAGE_MAP()
  721.  
  722. void CLocalComboBox::GetTheText(CString& str)
  723. {
  724.     int nIndex = GetCurSel();
  725.     if (nIndex == CB_ERR)
  726.         GetWindowText(str);
  727.     else
  728.         GetLBText(nIndex, str);
  729. }
  730.  
  731. void CLocalComboBox::SetTheText(LPCTSTR lpszText,BOOL bMatchExact)
  732. {
  733.     int idx = (bMatchExact) ? FindStringExact(-1,lpszText) : 
  734.         FindString(-1, lpszText);
  735.     SetCurSel( (idx==CB_ERR) ? -1 : idx);
  736.     if (idx == CB_ERR)
  737.         SetWindowText(lpszText);
  738. }
  739.  
  740. BOOL CLocalComboBox::LimitText(int nMaxChars)
  741. {
  742.     BOOL b = CComboBox::LimitText(nMaxChars);
  743.     if (b)
  744.         m_nLimitText = nMaxChars;
  745.     return b;
  746. }
  747.  
  748. int CLocalComboBox::OnCreate(LPCREATESTRUCT lpCreateStruct)
  749. {
  750.     if (CComboBox::OnCreate(lpCreateStruct) == -1)
  751.         return -1;
  752.       SendMessage(WM_SETFONT, (WPARAM)GetStockObject(theApp.m_nDefFont));
  753.     return 0;
  754. }
  755.  
  756. BOOL CLocalComboBox::PreTranslateMessage(MSG* pMsg)
  757. {
  758.     if (pMsg->message == WM_KEYDOWN)
  759.     {
  760.         CFormatBar* pBar = (CFormatBar*)GetParent();
  761.         switch (pMsg->wParam)
  762.         {
  763.         case VK_ESCAPE:
  764.             pBar->SyncToView();
  765.             pBar->NotifyOwner(NM_RETURN);
  766.             return TRUE;
  767.         case VK_RETURN:
  768.             pBar->NotifyOwner(NM_RETURN);
  769.             return TRUE;
  770.         case VK_TAB:
  771.             pBar->GetNextDlgTabItem(this)->SetFocus();
  772.             return TRUE;
  773.         case VK_UP:
  774.         case VK_DOWN:
  775.             if ((GetKeyState(VK_MENU) >= 0) && (GetKeyState(VK_CONTROL) >=0) && 
  776.                 !GetDroppedState())
  777.             {
  778.                 ShowDropDown();
  779.                 return TRUE;
  780.             }
  781.         }
  782.     }
  783.     return CComboBox::PreTranslateMessage(pMsg);
  784. }
  785.  
  786. void CFormatBar::NotifyOwner(UINT nCode)
  787. {
  788.     NMHDR nm;
  789.     nm.hwndFrom = m_hWnd;
  790.     nm.idFrom = GetDlgCtrlID();
  791.     nm.code = nCode;
  792.     GetOwner()->SendMessage(WM_NOTIFY, nm.idFrom, (LPARAM)&nm);
  793. }
  794.  
  795. void CFormatBar::SetCharFormat(CCharFormat& cf)
  796. {
  797.     CHARHDR fnm;
  798.     fnm.hwndFrom = m_hWnd;
  799.     fnm.idFrom = GetDlgCtrlID();
  800.     fnm.code = FN_SETFORMAT;
  801.     fnm.cf = cf;
  802.     VERIFY(GetOwner()->SendMessage(WM_NOTIFY, fnm.idFrom, (LPARAM)&fnm));
  803. }
  804.